home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
CU Amiga Super CD-ROM 15
/
CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso
/
CUCD
/
Graphics
/
Ghostscript
/
source
/
gsrefct.h
< prev
next >
Wrap
C/C++ Source or Header
|
1997-04-27
|
4KB
|
137 lines
/* Copyright (C) 1993, 1994, 1996, 1997 Aladdin Enterprises. All rights reserved.
This file is part of Aladdin Ghostscript.
Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND. No author
or distributor accepts any responsibility for the consequences of using it,
or for whether it serves any particular purpose or works at all, unless he
or she says so in writing. Refer to the Aladdin Ghostscript Free Public
License (the "License") for full details.
Every copy of Aladdin Ghostscript must include a copy of the License,
normally in a plain ASCII text file named PUBLIC. The License grants you
the right to copy, modify and redistribute Aladdin Ghostscript, but only
under certain conditions described in the License. Among other things, the
License requires that the copyright notice and this notice be preserved on
all copies.
*/
/* gsrefct.h */
/* Reference counting definitions */
#ifndef gsrefct_INCLUDED
# define gsrefct_INCLUDED
/*
* In many places below, a do {...} while (0) avoids problems with a possible
* enclosing 'if'.
*/
/*
* A reference-counted object must include the following header:
* rc_header rc;
* The header need not be the first element of the object.
*/
typedef struct rc_header_s rc_header;
struct rc_header_s {
long ref_count;
gs_memory_t *memory;
#define rc_free_proc(proc)\
void proc(P3(gs_memory_t *, void *, client_name_t))
rc_free_proc((*free));
};
/* ------ Allocate/free ------ */
rc_free_proc(rc_free_struct_only);
#define rc_alloc_struct_n(vp, typ, pstyp, mem, errstat, cname, rcinit)\
do\
{ if ( ((vp) = gs_alloc_struct(mem, typ, pstyp, cname)) == 0 ) {\
errstat;\
} else {\
(vp)->rc.ref_count = rcinit;\
(vp)->rc.memory = mem;\
(vp)->rc.free = rc_free_struct_only;\
}\
}\
while (0)
#define rc_alloc_struct_0(vp, typ, pstype, mem, errstat, cname)\
rc_alloc_struct_n(vp, typ, pstype, mem, errstat, cname, 0)
#define rc_alloc_struct_1(vp, typ, pstype, mem, errstat, cname)\
rc_alloc_struct_n(vp, typ, pstype, mem, errstat, cname, 1)
#define rc_free_struct(vp, cname)\
(*(vp)->rc.free)((vp)->rc.memory, (void *)(vp), cname)
/* ------ Reference counting ------ */
/* Increment a reference count. */
#define rc_increment(vp)\
do { if ( (vp) != 0 ) (vp)->rc.ref_count++; } while (0)
/* Increment a reference count, allocating the structure if necessary. */
#define rc_allocate_struct(vp, typ, pstype, mem, errstat, cname)\
do\
{ if ( (vp) != 0 )\
(vp)->rc.ref_count++;\
else\
rc_alloc_struct_1(vp, typ, pstype, mem, errstat, cname);\
}\
while (0)
/* Guarantee that a structure is allocated and is not shared. */
#define rc_unshare_struct(vp, typ, pstype, mem, errstat, cname)\
do\
{ if ( (vp) == 0 || (vp)->rc.ref_count > 1 || (vp)->rc.memory != (mem) )\
{ typ *new;\
rc_alloc_struct_1(new, typ, pstype, mem, errstat, cname);\
if ( vp ) (vp)->rc.ref_count--;\
(vp) = new;\
}\
}\
while (0)
/* Adjust a reference count either up or down. */
#define rc_adjust_(vp, delta, cname, body)\
do\
{ if ( (vp) != 0 && !((vp)->rc.ref_count += delta) )\
{ rc_free_struct(vp, cname);\
body;\
}\
}\
while (0)
#define rc_adjust(vp, delta, cname)\
rc_adjust_(vp, delta, cname, (vp) = 0)
#define rc_adjust_only(vp, delta, cname)\
rc_adjust_(vp, delta, cname, DO_NOTHING)
#define rc_adjust_const(vp, delta, cname)\
rc_adjust_only(vp, delta, cname)
#define rc_decrement(vp, cname)\
rc_adjust(vp, -1, cname)
#define rc_decrement_only(vp, cname)\
rc_adjust_only(vp, -1, cname)
/* Assign a pointer, adjusting reference counts. */
#define rc_assign(vpto, vpfrom, cname)\
do\
{ if ( (vpto) != (vpfrom) )\
{ rc_decrement_only(vpto, cname);\
(vpto) = (vpfrom);\
rc_increment(vpto);\
}\
}\
while (0)
/* Adjust reference counts for assigning a pointer, */
/* but don't do the assignment. We use this before assigning */
/* an entire structure containing reference-counted pointers. */
#define rc_pre_assign(vpto, vpfrom, cname)\
do\
{ if ( (vpto) != (vpfrom) )\
{ rc_decrement_only(vpto, cname);\
rc_increment(vpfrom);\
}\
}\
while (0)
#endif /* gsrefct_INCLUDED */